home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
Software Vault: The Gold Collection
/
Software Vault - The Gold Collection (American Databankers) (1993).ISO
/
cdr01
/
halcn305.zip
/
GSOB_DTE.PAS
< prev
next >
Wrap
Pascal/Delphi Source File
|
1993-05-02
|
12KB
|
439 lines
unit GSOB_Dte;
{-----------------------------------------------------------------------------
Date Processor
GSOB_DTE Copyright (c) Richard F. Griffin
02 April 1993
102 Molded Stone Pl
Warner Robins, GA 31088
-------------------------------------------------------------
This unit handles date conversion.
SHAREWARE -- COMMERCIAL USE RESTRICTED
Changes:
07 Feb 93 - Added GS_Date_CurCentury to return the current
century (i.e., '19' for 1992) based on the date
in the computer. This replaced the embedded
constant '19' found before. This is just in case
the routine is needed after the year 2000!
Acknowledgements:
An astronomers' Julian day number is a calendar system which is useful
over a very large span of time. (January 1, 1988 A.D. is 2,447,162 in
this system.) The mathematics of these procedures originally restricted
the valid range to March 1, 0000 through February 28, 4000. The update
by Carley Phillips changes the valid end date to December 31, 65535.
The basic algorithms are based on those contained in the COLLECTED
ALGORITHMS from Communications of the ACM, algorithm number 199,
originally submitted by Robert G. Tantzen in the August, 1963 issue
(Volume 6, Number 8). Note that these algorithms do not take into
account that years divisible by 4000 are NOT leap years. Therefore the
calculations are only valid until 02-28-4000. These procedures were
modified by Carley Phillips (76630,3312) to provide a mathematically
valid range of 03-01-0000 through 12-31-65535.
The main part of Tantzen's original algorithm depends on treating
January and February as the last months of the preceding year. Then,
one can look at a series of four years (for example, 3-1-84 through
2-29-88) in which the last day will be either the 1460th or the 1461st
day depending on whether the 4-year series ended in a leap day.
By assigning a longint julian date, computing differences between
dates, adding days to an existing date, and other mathematical actions
become much easier.
------------------------------------------------------------------------------}
{$O+}
interface
uses
{$IFDEF WINDOWS}
WinDOS;
{$ELSE}
DOS;
{$ENDIF}
const
GS_Date_JulInv = -1; {constant for invalid Julian day}
type
GS_Date_StrTyp = string[10];
GS_Date_ValTyp = longint;
GS_Date_CenTyp = string[2];
DateCountry = (American,ANSI,British,French,German,Italian,Japan,
USA, MDY, DMY, YMD);
var
GS_Date_Century : boolean;
GS_Date_Type : DateCountry;
function GS_Date_CurCentury : GS_Date_CenTyp;
function GS_Date_Curr : GS_Date_ValTyp;
function GS_Date_DBStor(nv : GS_Date_ValTyp) : GS_Date_StrTyp;
function GS_Date_View(nv : GS_Date_ValTyp) : GS_Date_StrTyp;
function GS_Date_Juln(sdate : GS_Date_StrTyp) : GS_Date_ValTyp;
function GS_Date_MDY2Jul(month, day, year : word) : GS_Date_ValTyp;
procedure GS_Date_Jul2MDY(jul : GS_Date_ValTyp; var month, day, year : word);
implementation
const
JulianConstant = 1721119; {constant for Julian day for 02-28-0000}
JulianMin = 1721120; {constant for Julian day for 03-01-0000}
JulianMax = 25657575; {constant for Julian day for 12-31-65535}
ThisCentury : GS_Date_CenTyp = '';
type
Str4 = string[4];
function DateType_MDY(mm, dd, yy: Str4): GS_Date_StrTyp;
var
ss : string[10];
begin
case GS_Date_Type of
American,
MDY : ss := ' / / ';
USA : ss := ' - - ';
end;
if GS_Date_Century then ss[0] := #10 else ss[0] := #8;
if mm <> '' then
begin
move(mm[1],ss[1],2);
move(dd[1],ss[4],2);
if GS_Date_Century then
move(yy[1],ss[7],4)
else
move(yy[3],ss[7],2);
end;
DateType_MDY := ss;
end;
function DateType_DMY(mm, dd, yy: Str4): GS_Date_StrTyp;
var
ss : string[10];
begin
case GS_Date_Type of
British,
French,
DMY : ss := ' / / ';
German : ss := ' . . ';
Italian : ss := ' - - ';
end;
if GS_Date_Century then ss[0] := #10 else ss[0] := #8;
if mm <> '' then
begin
move(dd[1],ss[1],2);
move(mm[1],ss[4],2);
if GS_Date_Century then
move(yy[1],ss[7],4)
else
move(yy[3],ss[7],2);
end;
DateType_DMY := ss;
end;
function DateType_YMD(mm, dd, yy: Str4): GS_Date_StrTyp;
var
ss : string[10];
begin
case GS_Date_Type of
Japan,
YMD : ss := ' / / ';
ANSI : ss := ' . . ';
end;
if not GS_Date_Century then system.Delete(ss,1,2);
if mm <> '' then
begin
if GS_Date_Century then
begin
move(yy[1],ss[1],4);
move(mm[1],ss[6],2);
move(dd[1],ss[9],2);
end
else
begin
move(yy[3],ss[1],2);
move(mm[1],ss[4],2);
move(dd[1],ss[7],2);
end;
end;
DateType_YMD := ss;
end;
function LeapYearTrue (year : word) : boolean;
begin
LeapYearTrue := false;
if (year mod 4 = 0) then
if (year mod 100 <> 0) or (year mod 400 = 0) then
if (year mod 4000 <> 0) then
LeapYearTrue := true;
end;
function DateOk (month, day, year : word) : boolean;
var
daz : integer;
begin
if (day <> 0) and
((month > 0) and (month < 13)) and
((year <> 0) or (month > 2)) then
begin
case month of
2 : begin
daz := 28;
if (LeapYearTrue(year)) then inc(daz);
end;
4,
6,
9,
11 : daz := 30;
else daz := 31;
end;
DateOk := day <= daz;
end
else DateOk := false;
end;
function GS_Date_MDY2Jul(month, day, year : word) : GS_Date_ValTyp;
var
wmm,
wyy,
jul : longint;
begin
wyy := year;
if (month > 2) then wmm := month - 3
else
begin
wmm := month + 9;
dec(wyy);
end;
jul := (wyy div 4000) * 1460969;
wyy := (wyy mod 4000);
jul := jul +
(((wyy div 100) * 146097) div 4) +
(((wyy mod 100) * 1461) div 4) +
(((153 * wmm) + 2) div 5) +
day +
JulianConstant;
if (jul < JulianMin) or (JulianMax < jul) then
jul := GS_Date_JulInv;
GS_Date_MDY2Jul := jul;
end;
procedure GS_Date_Jul2MDY(jul : GS_Date_ValTyp; var month, day, year : word);
var
tmp1 : longint;
tmp2 : longint;
begin
if (JulianMin <= jul) and (jul <= JulianMax) then
begin
tmp1 := jul - JulianConstant; {will be at least 1}
year := ((tmp1-1) div 1460969) * 4000;
tmp1 := ((tmp1-1) mod 1460969) + 1;
tmp1 := (4 * tmp1) - 1;
tmp2 := (4 * ((tmp1 mod 146097) div 4)) + 3;
year := (100 * (tmp1 div 146097)) + (tmp2 div 1461) + year;
tmp1 := (5 * (((tmp2 mod 1461) + 4) div 4)) - 3;
month := tmp1 div 153;
day := ((tmp1 mod 153) + 5) div 5;
if (month < 10) then
month := month + 3
else
begin
month := month - 9;
year := year + 1;
end {else}
end {if}
else
begin
month := 0;
day := 0;
year := 0;
end; {else}
end;
function GS_Date_CurCentury : GS_Date_CenTyp;
Var
month, day, year : word;
cw : word;
begin
if ThisCentury = '' then
begin
GetDate(year,month,day,cw);
year := year div 100;
Str(year:2, ThisCentury);
end;
GS_Date_CurCentury := ThisCentury
end;
function GS_Date_Curr : GS_Date_ValTyp;
Var
month, day, year : word;
cw : word;
begin
GetDate(year,month,day,cw);
GS_Date_Curr := GS_Date_MDY2Jul(month, day, year);
end;
function GS_Date_DBStor(nv : GS_Date_ValTyp) : GS_Date_StrTyp;
var
mm,
dd,
yy : word;
ss : string[8];
sg : string[4];
i : integer;
begin
ss := ' ';
if nv > GS_Date_JulInv then
begin
GS_Date_Jul2MDY(nv,mm,dd,yy);
str(mm:2,sg);
move(sg[1],ss[5],2);
str(dd:2,sg);
move(sg[1],ss[7],2);
str(yy:4,sg);
move(sg[1],ss[1],4);
for i := 1 to 8 do if ss[i] = ' ' then ss[i] := '0';
end;
GS_Date_DBStor := ss;
end;
function GS_Date_View(nv : GS_Date_ValTyp) : GS_Date_StrTyp;
var
mm,
dd,
yy : word;
ss : string[10];
sg1,
sg2,
sg3 : string[4];
i : integer;
begin
if nv > GS_Date_JulInv then
begin
GS_Date_Jul2MDY(nv,mm,dd,yy);
if mm = 0 then sg1 := ''
else
begin
str(mm:2,sg1);
str(dd:2,sg2);
str(yy:4,sg3);
end;
end else sg1 := '';
case GS_Date_Type of
American,
USA,
MDY : ss := DateType_MDY(sg1,sg2,sg3);
British,
French,
German,
Italian,
DMY : ss := DateType_DMY(sg1,sg2,sg3);
ANSI,
Japan,
YMD : ss := DateType_YMD(sg1,sg2,sg3);
end;
if sg1 <> '' then
for i := 1 to length(ss) do if ss[i] = ' ' then ss[i] := '0';
GS_Date_View := ss;
end;
function GS_Date_Juln(sdate : GS_Date_StrTyp) : GS_Date_ValTyp;
var
t : string[10];
valu,
yy,
mm,
dd : string[4];
mmn,
ddn,
yyn : word;
i : integer;
rsl : integer;
okDate : boolean;
co : longint;
begin
mm:= '';
dd := '';
yy := '';
t := sdate;
rsl := 0;
for i := length(t) downto 1 do
if t[i] < '0' then rsl := i;
if rsl = 0 then
begin
mm := copy(t,5,2);
dd := copy(t,7,2);
yy := copy(t,1,4);
end
else
begin
case GS_Date_Type of
American,
USA,
MDY : begin
mm := copy(t,1,2);
dd := copy(t,4,2);
yy := copy(t,7,4);
end;
British,
French,
German,
Italian,
DMY : begin
dd := copy(t,1,2);
mm := copy(t,4,2);
yy := copy(t,7,4);
end;
Japan,
YMD : begin
yy := copy(t,1,rsl-1);
mm := copy(t,rsl+1,2);
dd := copy(t,rsl+4,2);
end;
end;
if length(yy) = 2 then yy := GS_Date_CurCentury+yy;
end;
okDate := false;
val(mm,mmn,rsl);
if rsl = 0 then
begin
val(dd,ddn,rsl);
if rsl = 0 then
begin
val(yy,yyn,rsl);
if rsl = 0 then
begin
if DateOk(mmn,ddn,yyn) then okDate := true;
end;
end;
end;
if not okDate then
co := GS_Date_JulInv
else
begin
co := GS_Date_MDY2Jul(mmn, ddn, yyn);
end;
GS_Date_Juln := co;
end;
begin
GS_Date_Century := false;
GS_Date_Type := American;
end.